captcha_util.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. import base64
  2. import random
  3. import string
  4. from io import BytesIO
  5. from PIL import Image, ImageDraw, ImageFont
  6. from app.config.setting import settings
  7. class CaptchaUtil:
  8. """
  9. 验证码工具类
  10. """
  11. @classmethod
  12. def generate_captcha(cls) -> tuple[str, str]:
  13. """
  14. 生成带有噪声和干扰的验证码图片(4位随机字符)。
  15. 返回:
  16. - tuple[str, str]: Base64 PNG 字符串与验证码明文。
  17. """
  18. # 生成4位随机验证码
  19. chars = string.digits + string.ascii_letters
  20. captcha_value = "".join(random.sample(chars, 4))
  21. # 创建一张随机颜色背景的图片
  22. width, height = 160, 60
  23. background_color = tuple(random.randint(230, 255) for _ in range(3))
  24. image = Image.new("RGB", (width, height), color=background_color)
  25. draw = ImageDraw.Draw(image)
  26. # 使用指定字体
  27. font = ImageFont.truetype(font=settings.CAPTCHA_FONT_PATH, size=settings.CAPTCHA_FONT_SIZE)
  28. # 计算文本总宽度和高度
  29. total_width = sum(draw.textbbox((0, 0), char, font=font)[2] for char in captcha_value)
  30. text_height = draw.textbbox((0, 0), captcha_value[0], font=font)[3]
  31. # 计算起始位置,使文字居中
  32. x_start = (width - total_width) / 2
  33. y_start = (height - text_height) / 2 - draw.textbbox((0, 0), captcha_value[0], font=font)[1]
  34. # 绘制字符
  35. x = x_start
  36. for char in captcha_value:
  37. # 使用深色文字,增加对比度
  38. text_color = tuple(random.randint(0, 80) for _ in range(3))
  39. # 随机偏移,增加干扰
  40. x_offset = x + random.uniform(-2, 2)
  41. y_offset = y_start + random.uniform(-2, 2)
  42. # 绘制字符
  43. draw.text((x_offset, y_offset), char, font=font, fill=text_color)
  44. # 更新x坐标,增加字符间距的随机性
  45. x += draw.textbbox((0, 0), char, font=font)[2] + random.uniform(1, 5)
  46. # 添加干扰线
  47. for _ in range(4):
  48. line_color = tuple(random.randint(150, 200) for _ in range(3))
  49. points = [(i, int(random.uniform(0, height))) for i in range(0, width, 20)]
  50. draw.line(points, fill=line_color, width=1)
  51. # 添加随机噪点
  52. for _ in range(width * height // 60):
  53. point_color = tuple(random.randint(0, 255) for _ in range(3))
  54. draw.point(
  55. (random.randint(0, width), random.randint(0, height)),
  56. fill=point_color,
  57. )
  58. # 将图像数据保存到内存中并转换为base64
  59. buffer = BytesIO()
  60. image.save(buffer, format="PNG", optimize=True)
  61. base64_string = base64.b64encode(buffer.getvalue()).decode()
  62. return base64_string, captcha_value
  63. @classmethod
  64. def captcha_arithmetic(cls, difficulty: str = "medium") -> tuple[str, int]:
  65. """
  66. 创建算术验证码图片(加减乘运算);浅色底、居中算式,无旋转与干扰线/噪点。
  67. 参数:
  68. - difficulty (str): 难度级别(easy / medium / hard),控制数字范围与可用运算符。
  69. 返回:
  70. - tuple[str, int]: base64 编码的 PNG 图片字符串与正确答案(整数)。
  71. """
  72. difficulty_config = {
  73. "easy": {"num_range": (1, 9), "operators": ["+", "-"]},
  74. "medium": {"num_range": (1, 15), "operators": ["+", "-", "*"]},
  75. "hard": {"num_range": (1, 20), "operators": ["+", "-", "*"]},
  76. }
  77. config = difficulty_config.get(difficulty, difficulty_config["medium"])
  78. operators = config["operators"]
  79. operator = random.choice(operators)
  80. num_range = config["num_range"]
  81. if operator == "-":
  82. num1 = random.randint(num_range[0] + 5, num_range[1])
  83. num2 = random.randint(num_range[0], num1 - 1)
  84. elif operator == "*":
  85. num1 = random.randint(num_range[0], min(10, num_range[1]))
  86. num2 = random.randint(num_range[0], min(10, num_range[1]))
  87. else:
  88. num1 = random.randint(num_range[0], num_range[1])
  89. num2 = random.randint(num_range[0], num_range[1])
  90. result_map = {
  91. "+": lambda x, y: x + y,
  92. "-": lambda x, y: x - y,
  93. "*": lambda x, y: x * y,
  94. }
  95. captcha_value = result_map[operator](num1, num2)
  96. width, height = 160, 60
  97. image = Image.new("RGB", (width, height), color=(248, 249, 250))
  98. draw = ImageDraw.Draw(image)
  99. font = ImageFont.truetype(font=settings.CAPTCHA_FONT_PATH, size=settings.CAPTCHA_FONT_SIZE)
  100. text = f"{num1} {operator} {num2} = ?"
  101. tb = draw.textbbox((0, 0), text, font=font)
  102. tw, th = tb[2] - tb[0], tb[3] - tb[1]
  103. x = (width - tw) // 2
  104. y = (height - th) // 2 - tb[1]
  105. draw.text((x, y), text, fill=(55, 65, 81), font=font)
  106. buffer = BytesIO()
  107. image.save(buffer, format="PNG", optimize=True)
  108. base64_string = base64.b64encode(buffer.getvalue()).decode()
  109. return base64_string, captcha_value